diff options
| author | Armand Philippot <git@armandphilippot.com> | 2022-05-15 17:45:41 +0200 | 
|---|---|---|
| committer | Armand Philippot <git@armandphilippot.com> | 2022-05-15 19:06:42 +0200 | 
| commit | c95cce04393080a52a54191cff7be8fec68af4b0 (patch) | |
| tree | 1022bc574c8fc8e657be922b26c1cf16cbfd9071 /src/pages/article/[slug].tsx | |
| parent | 235fe67d770f83131c9ec10b99012319440db690 (diff) | |
chore: add Article pages
Diffstat (limited to 'src/pages/article/[slug].tsx')
| -rw-r--r-- | src/pages/article/[slug].tsx | 251 | 
1 files changed, 251 insertions, 0 deletions
| diff --git a/src/pages/article/[slug].tsx b/src/pages/article/[slug].tsx new file mode 100644 index 0000000..6a47c16 --- /dev/null +++ b/src/pages/article/[slug].tsx @@ -0,0 +1,251 @@ +import ButtonLink from '@components/atoms/buttons/button-link'; +import Link from '@components/atoms/links/link'; +import { type BreadcrumbItem } from '@components/molecules/nav/breadcrumb'; +import Sharing from '@components/organisms/widgets/sharing'; +import PageLayout, { +  type PageLayoutProps, +} from '@components/templates/page/page-layout'; +import { +  getAllArticlesSlugs, +  getArticleBySlug, +} from '@services/graphql/articles'; +import { getPostComments } from '@services/graphql/comments'; +import { type Article, type Comment } from '@ts/types/app'; +import { loadTranslation, type Messages } from '@utils/helpers/i18n'; +import useSettings from '@utils/hooks/use-settings'; +import { GetStaticPaths, GetStaticProps, NextPage } from 'next'; +import Head from 'next/head'; +import { useRouter } from 'next/router'; +import Script from 'next/script'; +import { ParsedUrlQuery } from 'querystring'; +import { useIntl } from 'react-intl'; +import { Blog, BlogPosting, Graph, WebPage } from 'schema-dts'; +import useSWR from 'swr'; + +type ArticlePageProps = { +  comments: Comment[]; +  post: Article; +  translation: Messages; +}; + +/** + * Article page. + */ +const ArticlePage: NextPage<ArticlePageProps> = ({ comments, post }) => { +  const { content, id, intro, meta, slug, title } = post; +  const { author, commentsCount, cover, dates, seo, thematics, topics } = meta; +  const { data } = useSWR(() => id, getPostComments, { +    fallbackData: comments, +  }); +  const intl = useIntl(); +  const homeLabel = intl.formatMessage({ +    defaultMessage: 'Home', +    description: 'Breadcrumb: home label', +    id: 'j5k9Fe', +  }); +  const blogLabel = intl.formatMessage({ +    defaultMessage: 'Blog', +    description: 'Breadcrumb: blog label', +    id: 'Es52wh', +  }); +  const breadcrumb: BreadcrumbItem[] = [ +    { id: 'home', name: homeLabel, url: '/' }, +    { id: 'blog', name: blogLabel, url: '/blog' }, +    { id: 'article', name: title, url: `/article/${slug}` }, +  ]; + +  const headerMeta: PageLayoutProps['headerMeta'] = { +    author: author?.name, +    publication: { date: dates.publication }, +    update: dates.update ? { date: dates.update } : undefined, +    thematics: +      thematics && +      thematics.map((thematic) => ( +        <Link key={thematic.id} href={`/thematique/${thematic.slug}`}> +          {thematic.name} +        </Link> +      )), +  }; + +  const footerMeta: PageLayoutProps['footerMeta'] = { +    topics: +      topics && +      topics.map((topic) => { +        return ( +          <ButtonLink key={topic.id} target={`/sujet/${topic.slug}`}> +            {topic.name} +          </ButtonLink> +        ); +      }), +  }; + +  const { website } = useSettings(); +  const { asPath } = useRouter(); +  const pageUrl = `${website.url}${asPath}`; +  const pagePublicationDate = new Date(dates.publication); +  const pageUpdateDate = dates.update ? new Date(dates.update) : undefined; + +  const webpageSchema: WebPage = { +    '@id': `${pageUrl}`, +    '@type': 'WebPage', +    breadcrumb: { '@id': `${website.url}/#breadcrumb` }, +    lastReviewed: dates.update, +    name: seo.title, +    description: seo.description, +    reviewedBy: { '@id': `${website.url}/#branding` }, +    url: `${pageUrl}`, +    isPartOf: { +      '@id': `${website.url}`, +    }, +  }; + +  const blogSchema: Blog = { +    '@id': `${website.url}/#blog`, +    '@type': 'Blog', +    blogPost: { '@id': `${website.url}/#article` }, +    isPartOf: { +      '@id': `${pageUrl}`, +    }, +    license: 'https://creativecommons.org/licenses/by-sa/4.0/deed.fr', +  }; + +  const blogPostSchema: BlogPosting = { +    '@id': `${website.url}/#article`, +    '@type': 'BlogPosting', +    name: title, +    description: intro, +    articleBody: content, +    author: { '@id': `${website.url}/#branding` }, +    commentCount: commentsCount, +    copyrightYear: pagePublicationDate.getFullYear(), +    creator: { '@id': `${website.url}/#branding` }, +    dateCreated: pagePublicationDate.toISOString(), +    dateModified: pageUpdateDate && pageUpdateDate.toISOString(), +    datePublished: pagePublicationDate.toISOString(), +    discussionUrl: `${pageUrl}/#comments`, +    editor: { '@id': `${website.url}/#branding` }, +    headline: title, +    image: cover?.src, +    inLanguage: website.locales.default, +    isPartOf: { +      '@id': `${website.url}/blog`, +    }, +    license: 'https://creativecommons.org/licenses/by-sa/4.0/deed.fr', +    mainEntityOfPage: { '@id': `${pageUrl}` }, +    thumbnailUrl: cover?.src, +  }; + +  const schemaJsonLd: Graph = { +    '@context': 'https://schema.org', +    '@graph': [webpageSchema, blogSchema, blogPostSchema], +  }; + +  /** +   * Convert the comments list to the right format. +   * +   * @param {Comment[]} list - The comments list. +   * @returns {PageLayoutProps['comments']} - The formatted comments list. +   */ +  const getCommentsList = (list: Comment[]): PageLayoutProps['comments'] => { +    return list.map((comment) => { +      const { +        content: commentBody, +        id: commentId, +        meta: commentMeta, +        parentId, +        replies, +      } = comment; +      const { author: commentAuthor, date } = commentMeta; +      const { name, avatar, website: authorUrl } = commentAuthor; + +      return { +        author: { name, avatar: avatar!.src, url: authorUrl }, +        content: commentBody, +        id: commentId, +        publication: date, +        child: getCommentsList(replies), +        parentId, +      }; +    }); +  }; + +  return ( +    <> +      <Head> +        <title>{seo.title}</title> +        <meta name="description" content={seo.description} /> +        <meta property="og:url" content={`${pageUrl}`} /> +        <meta property="og:type" content="article" /> +        <meta property="og:title" content={title} /> +        <meta property="og:description" content={intro} /> +      </Head> +      <Script +        id="schema-project" +        type="application/ld+json" +        dangerouslySetInnerHTML={{ __html: JSON.stringify(schemaJsonLd) }} +      /> +      <PageLayout +        allowComments={true} +        breadcrumb={breadcrumb} +        comments={data && getCommentsList(data)} +        footerMeta={footerMeta} +        headerMeta={headerMeta} +        id={id as number} +        intro={intro} +        title={title} +        withToC={true} +        widgets={[ +          <Sharing +            key="sharing-widget" +            data={{ excerpt: intro, title, url: pageUrl }} +            media={[ +              'diaspora', +              'email', +              'facebook', +              'journal-du-hacker', +              'linkedin', +              'twitter', +            ]} +          />, +        ]} +      > +        {content} +      </PageLayout> +    </> +  ); +}; + +interface PostParams extends ParsedUrlQuery { +  slug: string; +} + +export const getStaticProps: GetStaticProps<ArticlePageProps> = async ({ +  locale, +  params, +}) => { +  const post = await getArticleBySlug(params!.slug as PostParams['slug']); +  const comments = await getPostComments(post.id as number); +  const translation = await loadTranslation(locale); + +  return { +    props: { +      comments: JSON.parse(JSON.stringify(comments)), +      post: JSON.parse(JSON.stringify(post)), +      translation, +    }, +  }; +}; + +export const getStaticPaths: GetStaticPaths = async () => { +  const slugs = await getAllArticlesSlugs(); +  const paths = slugs.map((slug) => { +    return { params: { slug } }; +  }); + +  return { +    paths, +    fallback: true, +  }; +}; + +export default ArticlePage; | 
